node学习笔记——初识Express

Express 在后台的受欢迎的程度,和原来的jQuery一样,就是企业的事实上的标准。

为什么需要 Express 呢?因为原生Node开发,会发现有很多问题。比如:

  • 呈递静态页面很不方便,需要处理每个HTTP请求,还要考虑304问题
  • 路由处理代码不直观清晰,需要写很多正则表达式和字符串函数
  • 不能集中精力写业务,要考虑很多其他的东西

Express 的哲学是在你的想法和服务器之间充当薄薄的一层。这并不意味着他不够健壮,或者没有足够的有用特性,而是尽量少干预你,让你充分表达自己的思想,同时提供一些有用的东西。

基本功能

路由能力:

Express有着惊艳的路由能力,正则提取数据能力足够一般工作使用;

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
var express = require("express");
var app = express();

app.get("/",function(req,res){
res.send("你好");
});

app.get("/haha",function(req,res){
res.send("这是haha页面,哈哈哈哈哈哈");
});

app.get(/^\/student\/([\d]{10})$/,function(req,res){
res.send("学生信息,学号" + req.params[0]);
});

app.get("/teacher/:gonghao",function(req,res){
res.send("老师信息,工号" + req.params.gonghao);
});

app.listen(3000);

静态化文件能力:

Express中静态文件处理,就是一句话的事儿:

1
2
3
4
5
6
7
var express = require("express");

var app = express();

app.use(express.static("./public"));

app.listen(3000);

模板引擎

Express与模板引擎的配合,直观清晰:

1
2
3
4
5
6
7
8
9
10
11
12
13
var express = require("express");
var app = express();

//设置ejs模板引擎
app.set("view engine","ejs");

app.get("/",function(req,res){
res.render("haha",{ //haha为目标文件名:haha.ejs
"news" : ["我是小新闻啊","我也是啊","哈哈哈哈"]
});
});

app.listen(3000);

路由

当用get请求访问一个网址的时候,做什么事情:

1
2
3
app.get("网址",function(req,res){

});

当用post访问一个网址的时候,做什么事情:

1
2
3
app.post("网址",function(req,res){

});

如果想处理这个网址的任何method的请求,那么写all:

1
2
3
app.all("/",function(){

});

这里的网址,不分大小写,也就是说,你路由是:

1
2
3
app.get("/AAb",function(req,res){
res.send("你好");
});

实际上小写的访问也行。所有的GET参数,? 后面的都已经被忽略。 锚点#也被忽略,你路由到/a ,实际/a?id=2&sex=nan 也能被处理。

正则表达式可以被使用。正则表达式中,未知部分用圆括号分组,然后可用req.params[0]、[1]得到。req.params类数组对象。

1
2
3
app.get(/^\/student\/([\d]{10})$/,function(req,res){
res.send("学生信息,学号" + req.params[0]);
});

冒号可以达到一样的传递获取变量的效果,冒号是更推荐的写法:

1
2
3
4
5
6
7
8
9
app.get("/student/:id",function(req,res){
var id = req.params["id"];
var reg= /^[\d]{6}$/;
if(reg.test(id)){
res.send(id);
}else{
res.send("请检查格式");
}
});

表单可以自己提交到自己上:

1
2
3
4
5
6
7
8
app.get("/",function(req,res){
res.render("form");
});

app.post("/",function(req,res){
//将数据添加进入数据库
res.send("成功");
});

适合进行 RESTful路由设计。简单说,就是一个路径,但是http method不同,对这个页面的使用也不同。

服务器开启前的代码只执行一次:

1
2
3
4
5
6
7
8
9
10
11
12
13
var http = require("http");

// 这个语句,就在打开服务器的时候,执行一次。
// 每次用户访问的时候,不执行这个语句
var a = 100;

var server = http.createServer(function(req,res){
// 用户访问的时候,执行这里的语句:
a++;
res.end(a.toString());
});

server.listen(3000,"127.0.0.1");

中间件

如果我的的get、post回调函数中,没有next参数,那么就匹配上第一个路由,就不会往下匹配了。如果想往下匹配的话,那么需要写next()

1
2
3
4
5
6
7
8
app.get("/",function(req,res,next){
console.log("1");
next();
});

app.get("/",function(req,res){
console.log("2");
});

下面两个路由,感觉没有关系:

1
2
3
4
5
6
7
8
9
app.get("/:username/:id",function(req,res){
console.log("1");
res.send("用户信息" + req.params.username);
});

app.get("/admin/login",function(req,res){
console.log("2");
res.send("管理员登录");
});

但是实际上冲突了,因为admin可以当做用户名 login可以当做id。

解决方法1:交换位置。 也就是说,express中所有的路由(中间件)的顺序至关重要。

匹配上第一个,就不会往下匹配了。 具体的往上写,抽象的往下写:

1
2
3
4
5
6
7
8
9
app.get("/admin/login",function(req,res){
console.log("2");
res.send("管理员登录");
});

app.get("/:username/:id",function(req,res){
console.log("1");
res.send("用户信息" + req.params.username);
});

解决方法2:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
app.get("/:username/:id",function(req,res,next){
var username = req.params.username;
//检索数据库,如果username不存在,那么next()
if(检索数据库){
console.log("1");
res.send("用户信息");
}else{
next();
}
});

app.get("/admin/login",function(req,res){
console.log("2");
res.send("管理员登录");
});

路由get、post这些东西,就是中间件,中间件讲究顺序,匹配上第一个之后,就不会往后匹配了。next函数才能够继续往后匹配。

app.use()也是一个中间件。与get、post不同的是,他的网址不是精确匹配的。而是能够有小文件夹拓展的。比如网址: http://127.0.0.1:3000/admin/aa/bb/cc/dd

1
2
3
4
5
6
app.use("/admin",function(req,res){ 
res.write(req.originalUrl + "\n"); // /admin/aa/bb/cc/dd
res.write(req.baseUrl + "\n"); // /admin
res.write(req.path + "\n"); // /aa/bb/cc/dd
res.end("你好");
});

当你不写路径的时候,实际上就相当于”/“,就是所有网址

1
2
3
4
app.use(function(req,res,next){
console.log(new Date());
next();
});

所以,app.use()就给了我们增加一些特定功能的便利场所。实际上app.use()的东西,基本上都从第三方能得到。

EJS模板引擎

大多数情况下,渲染内容用res.render(),将会根据views中的模板文件进行渲染。如果不想使用views文件夹,想自己设置文件夹名字,那么 app.set("views","aaaa");

使用其他模板引擎:

1
2
3
4
5
6
7
8
9
10
11
12
13
var express = require("express");
var app = express();

//使用jade模板引擎,对应前端文件名为.jade结尾
app.engine('jade', require('jade').__express);
app.set("view engine","jade");

app.get("/",function(req,res){
console.log(req.ip);
res.render("xixi");
});

app.listen(3000);

如果想写一个快速测试页,当然可以使用res.send()。这个函数将根据内容,自动帮我们设置了Content-Type头部和200状态码。send()只能用一次,和end一样。和end不一样在哪里?能够自动设置MIME类型。

如果想使用不同的状态码,可以:

1
2
3
4
5
6
7
8
9
10
11
12
13
var express = require("express");
var app = express();

//静态服务
app.use("/jingtai",express.static("./public"));

//会自动识别err参数,如果有,那么就这个函数能捕获err
app.use(function(req,res){
//res.status设置不同的状态码
res.status(404).send("没有这个页面!");
});

app.listen(3000);

同样的,如果想使用不同的Content-Type,可以:

1
res.set('Content-Type', 'text/html');

GET请求和POST请求的参数

GET请求的参数在URL中,在原生Node中,需要使用url模块来识别参数字符串。在Express中,不需要使用url模块了。可以直接使用req.query对象:

1
2
3
4
5
6
7
8
9
var express = require("express");
var app = express();

app.get("/",function(req,res){
console.log(req.query);
res.send();
});

app.listen(3000);

POST请求在express中不能直接获得,必须使用body-parser模块。使用后,将可以用req.body得到参数。但是如果表单中含有文件上传,那么还是需要使用formidable模块。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
var express = require("express");
var bodyParser = require('body-parser')
var app = express();

//模板引擎
app.set("view engine","ejs");

app.get("/",function(req,res){
res.render("form");
});

//bodyParser API
app.use(bodyParser.urlencoded({ extended: false }))

app.post("/",function(req,res){
console.log(req.body);
});

app.listen(3000);



完~